-- Copyright 2017 Paul Kulchenko, ZeroBrane LLC; All rights reserved
-- Contributed by Chronos Phaenon Eosphoros (@cpeosphoros)
-- Some code taken from those plugins:
-- todo.lua:
-- 	Copyright 2014 Paul Kulchenko, ZeroBrane LLC; All rights reserved
-- 	Contributed by Mark Fainstein
-- analyzeall.lua:
-- 	Copyright 2014 Paul Kulchenko, ZeroBrane LLC; All rights reserved
-- 插件-代办列表（项目）
local id = ID("TODOAllpanel.referenceview")
local TODOpanel = "TODOAllpanel"
local refeditor
local spec = {iscomment = {}}

local function path2mask(s)
  return s
  :gsub('([%(%)%.%%%+%-%?%[%^%$%]])','%%%1') -- escape all special symbols
  :gsub("%*", ".*") -- but expand asterisk into sequence of any symbols
  :gsub("[\\/]","[\\\\/]") -- allow for any path
end

local fileTasks

local function mapTODOS(fileName, text)
  local i = 0
  local tasks = {}
  while true do
    --find next todo index
    i = string.find(text, "TODO"..":", i+1)
    if i == nil then
      fileTasks[fileName] = tasks
      break
    end
    local j = string.find(text, "\n",i+1)
    local taskStr = string.sub(text, i+5, j)
    table.insert(tasks, {pos = i, str = taskStr})
  end
end

local function readfile(filePath)
  local input = io.open(filePath)
  local data = input:read("*a")
  input:close()
  return data
end

local function sortedKeys(tbl)
  local sorted = {}
  for k, _ in pairs (tbl) do
    table.insert(sorted, k)
  end
  table.sort(sorted)
  return sorted
end

local projectPath

local function fileNameFromPath(filePath)
  return filePath:gsub(projectPath, "")
end

local function mapProject(self, editor)

  local specs = self:GetConfig().ignore or {}
  if editor then
    mapTODOS(fileNameFromPath(ide:GetDocument(editor):GetFilePath()), editor:GetText())
  else
    local masks = {}
    for i in ipairs(specs) do masks[i] = "^"..path2mask(specs[i]).."$" end
    for _, filePath in ipairs(ide:GetFileList(projectPath, true, "*.lua")) do
      local fileName = fileNameFromPath(filePath)
      local ignore = false or editor
      for _, spec in ipairs(masks) do
        ignore = ignore or fileName:find(spec)
      end
      -- TODO: testing here
      if not ignore then
        mapTODOS(fileName, readfile(filePath))
      end
    end
  end

  local files = sortedKeys(fileTasks)
  local tasksListStr = TR("TODO List")..": \n\n"
  local lineCounter = 2
  local positions = {}

  local function insertLine(line, pos, file)
    tasksListStr = tasksListStr .. line
    lineCounter = lineCounter + 1
    if pos then
      positions[lineCounter] = {pos = pos, file = file}
    end
  end

  for _, file in ipairs(files) do
    local tasks = fileTasks[file]
    if tasks and #tasks ~= 0 then
      insertLine(file .. ":\n", 1, file)
      for counter, taskStr in ipairs(tasks) do
        insertLine(counter..".["..taskStr.pos.."] "..taskStr.str, taskStr.pos, file)
      end
      insertLine("\n")
    end
  end

  refeditor:SetReadOnly(false)
  refeditor:SetText(tasksListStr)
  refeditor:SetReadOnly(true)

  --On click of a task, go to relevant position in the text
  refeditor:Connect(wxstc.wxEVT_STC_DOUBLECLICK, function()
      local line = refeditor:GetCurrentLine()+1
      local position = positions[line]
      if not position then return end
      local filePath = projectPath .. position.file
      local docs = ide:GetDocuments()

      local editor
      for _, doc in ipairs(docs) do
        if doc:GetFilePath() == filePath then
          editor = doc:GetEditor()
          break
        end
      end
      if not editor then
        editor = ide:LoadFile(filePath)
        if not editor then error("Couldn't load " .. filePath) end
      end

      editor:GotoPosEnforcePolicy(position.pos - 1)
      if not ide:GetEditorWithFocus(editor) then ide:GetDocument(editor):SetActive() end
    end)
end

return {
  name = "Show project-wide TODO panel",
  description = "Adds a project-wide panel for showing a tasks list.",
  author = "Chronos Phaenon Eosphoros",
  version = 0.13,
  dependencies = "1.60",

  onRegister = function(self)
    local e = ide:CreateBareEditor()
    refeditor = e

    local w, h = 250, 250
    local conf = function(pane)
      pane:Dock():MinSize(w,-1):BestSize(w,-1):FloatingSize(w,h)
    end
    local layout = ide:GetSetting("/view", "uimgrlayout")

    --if ide:IsPanelDocked(TODOpanel) then
    if not layout or not layout:find(TODOpanel) then
      ide:AddPanelDocked(ide.frame.projnotebook, e, TODOpanel, TR("TODO"), conf)
    else
      ide:AddPanel(e, TODOpanel, TR("TODO"), conf)
    end

    do -- markup handling in the reference panel
      -- copy some settings from the lua spec
      for _, s in ipairs({'lexer', 'lexerstyleconvert'}) do
        spec[s] = ide.specs.lua[s]
      end
      -- this allows the markup to be recognized in all token types
      for i = 0, 16 do spec.iscomment[i] = true end
      e:Connect(wxstc.wxEVT_STC_UPDATEUI, function() MarkupStyle(e,0,e:GetLineCount()) end)
    end

    e:SetReadOnly(true)
    e:SetWrapMode(wxstc.wxSTC_WRAP_NONE)
    e:SetupKeywords("lua",spec,ide:GetConfig().styles,ide:GetOutput():GetFont())

    -- remove all margins
    for m = 0, 4 do e:SetMarginWidth(m, 0) end

    -- disable dragging to the panel
    e:Connect(wxstc.wxEVT_STC_DO_DROP, function(event) event:SetDragResult(wx.wxDragNone) end)

    local menu = ide:GetMenuBar():GetMenu(ide:GetMenuBar():FindMenu(TR("&Project")))
    menu:InsertCheckItem(4, id, TR("TODO List")..KSC(id))
    menu:Connect(id, wx.wxEVT_COMMAND_MENU_SELECTED, function ()
        local uimgr = ide:GetUIManager()
        uimgr:GetPane(TODOpanel):Show(not uimgr:GetPane(TODOpanel):IsShown())
        uimgr:Update()
      end)
    ide:GetMainFrame():Connect(id, wx.wxEVT_UPDATE_UI, function (event)
        local pane = ide:GetUIManager():GetPane(TODOpanel)
        menu:Enable(event:GetId(), pane:IsOk()) -- disable if doesn't exist
        menu:Check(event:GetId(), pane:IsOk() and pane:IsShown())
      end)
  end,

  onUnRegister = function(self)
    ide:RemoveMenuItem(id)
  end,

  onProjectLoad = function(self, project)
    fileTasks = {}
    projectPath = project
    mapProject(self)
  end,

  onEditorCharAdded = function(self, editor) --, event)
    mapProject(self, editor)
  end,
}
